|
Pipelines exerciseIn this exercise, you create a model configuration based on the data in the file pipes.txt. Step 1Implement a function
that returns true, when the string given as a parameter is nonempty:
You need the empty string (!=) :: a -> a -> Boolean (Prelude) Step 2Implement a function
that removes a comment from a line. A comment starts with
The following functions are useful here: splitString :: String -> String -> [String] (Prelude)
(!) :: IndexedSequence a => a b -> Integer -> b (Prelude)
trim :: String -> String (Prelude) Removes leading and trailing whitespace from the string. Step 3New, lets read the file pipes.txt. Store it to somewhere in your file system. You need to import the module
Now, try to read the file using readLines :: String -> <Proc> [String] (StringIO) Reads all lines of the file whose name is given as a parameter. The file contents are expected to be UTF8 encoded. Remember that
Step 4As you see, the file contains empty lines and comments. Implement a function
that reads the file, whose name is given as a parameter, removes the comments, empty lines and leading and trailing
whitespace at every line. It returns the preprocessed lines. You need the
functions map :: FunctorE a => (b -> <d> c) -> a b -> <d> a c (Prelude) Applies the function to all elements of the container and returns the similarly shaped container with the results: For lists,
for example
filter :: MonadZeroE a => (b -> <c> Boolean) -> a b -> <c> a b (Prelude) Step 5Create a new SCL module and move your definitions there (if you have not done so already). It is much easier to continue handling the increasing number of function definitions there. Step 6You may have noticed that the lines in the preprocessed file have entries separated by
that check whether the first string in a list of strings is "POINT" or "PIPE". For example
Step 7Now, add the following definitions to your SCL module:
Create a function
that is called as
It should first read the file and preprocess it using flip :: (a -> b -> <d> c) -> b -> a -> <d> c (Prelude) Flips the parameters of a binary function. It should then filter the lines into two lists, one containing all definitions of points
and one all definitions of pipes. Finally, the function should call
iter :: FunctorE a => (b -> <d> c) -> a b -> <d> () (Prelude) Calls the given function with all elements of the given container. and iterI :: FunctorE a => (Integer -> b -> <d> c) -> a b -> <d> () (Prelude) Calls the given function with all elements of the given container giving also the index of the element as a parameter. When finished the function should work like this from the console:
Step 8Reimplement the function You need also the function read :: Read a => String -> a (Prelude) Converts a string to a required type of value. to convert the diagram coordinates from strings to doubles.
Add some prefix to the point indicies to form the point name (for example
Test your implementation with the example data. Step 9Reimplement the function Test your implementation again with the example data. Step 10In this final step, fix the coordinates of the pipes so that they are located between the points they connect. You may do this in the following way. In
that gives the coordinates of a point when given the index of the point. You may create it by partially applying the function index :: [(a, b)] -> a -> Maybe b (Prelude) Given a list of key-value pairs, the function produces a function that finds a value efficiently for the given key. for this. Then, add the new parameter
Now, you may read the coordinates of the points in
where fromJust :: Maybe a -> a (Prelude) Given Use the average of the connected points as the diagram coordinate for the pipe. |